<template>
  <div id="private-message">
    <div class="floating-header-shadow"></div>
    <div class="header-bar d-flex w-100">
      <!-- changing w-25 would also need changes in .left-header.w-25 -->
      <div class="d-flex left-header">
        <div
          v-once
          class="mail-icon svg-icon"
          v-html="icons.mail"
        ></div>
        <h2
          v-once
          class="flex-fill text-center mail-icon-label"
        >
          {{ $t('messages') }}
        </h2>

        <button
          class="btn btn-secondary plus-button"
          :class="{'new-message-mode':showStartNewConversationInput}"
          @click="triggerStartNewConversationState()"
        >
          <div
            class="svg-icon icon-10 color"
            v-html="icons.positive"
          ></div>
        </button>
      </div>

      <start-new-conversation-input-header
        v-if="showStartNewConversationInput"
        @startNewConversation="startConversationByUsername($event)"
        @cancelNewConversation="showStartNewConversationInput = false"
      />
      <div
        v-else-if="selectedConversation && selectedConversation.key"
        class="d-flex selected-conversion"
      >
        <router-link
          :to="{'name': 'userProfile', 'params': {'userId': selectedConversation.key}}"
        >
          <face-avatar
            v-if="selectedConversation.userStyles"
            :member="selectedConversation.userStyles"
            :class="selectedConversationFaceAvatarClass"
          />
        </router-link>
        <user-link
          :backer="selectedConversation.backer"
          :contributor="selectedConversation.contributor"
          :name="selectedConversation.name"
          :user="selectedConversation"
          :user-id="selectedConversation.key"
          :hide-tooltip="true"
        />
      </div>
    </div>
    <div class="d-flex content">
      <div class="sidebar d-flex flex-column">
        <div class="disable-background">
          <toggle-switch
            :label="optTextSet.switchDescription"
            :checked="user.inbox.optOut"
            :hover-text="optTextSet.popoverText"
            @change="toggleOpt()"
          />
        </div>
        <pm-conversations-list
          :filters-conversations="filtersConversations"
          :selected-conversation="selectedConversation"
          @selectConversation="selectConversation($event)"
        />
        <button
          v-if="canLoadMoreConversations"
          class="btn btn-secondary"
          @click="loadConversations()"
        >
          {{ $t('loadMore') }}
        </button>
      </div>

      <div class="messages-column d-flex flex-column align-items-center">
        <div
          v-if="user.inbox.optOut"
          class="disable-background-in-message-list"
        >
          <span
            v-once
            class="caption"
          > {{ $t('PMDisabledCaptionTitle') }}. </span> &nbsp;
          <span
            v-once
            class="text"
          > {{ $t('PMDisabledCaptionText') }} </span>
        </div>

        <pm-empty-state
          v-if="uiState === UI_STATES.NO_CONVERSATIONS"
          :chat-revoked="user.flags.chatRevoked"
          @newMessageClicked="showStartNewConversationInput = true"
        />

        <pm-new-message-started
          v-if="uiState === UI_STATES.START_NEW_CONVERSATION && selectedConversation.userStyles"
          :member-obj="selectedConversation.userStyles"
        />

        <div
          v-if="uiState === UI_STATES.NO_CONVERSATIONS_SELECTED"
          class="empty-messages full-height m-auto text-center"
        >
          <div class="no-messages-box">
            <div
              v-once
              class="svg-icon envelope"
              v-html="icons.messageIcon"
            ></div>
            <h2>{{ placeholderTexts.title }}</h2>
            <p v-html="placeholderTexts.description"></p>
          </div>
        </div>

        <messageList
          v-if="selectedConversation && selectedConversationMessages.length > 0"
          ref="chatscroll"
          class="message-scroll"
          :chat="selectedConversationMessages"
          :conversation-opponent-user="selectedConversation.userStyles"
          :can-load-more="canLoadMore"
          :is-loading="messagesLoading"
          @message-removed="messageRemoved"
          @message-liked="messageLiked"
          @triggerLoad="infiniteScrollTrigger"
        />

        <pm-disabled-state
          v-if="disabledTexts?.showBottomInfo"
          :disabled-texts="disabledTexts"
        />
        <div
          v-if="shouldShowInputPanel"
          class="full-width"
        >
          <div
            class="new-message-row d-flex align-items-center"
          >
            <textarea
              ref="textarea"
              v-model="newMessage"
              dir="auto"
              class="flex-fill"
              :placeholder="$t('needsTextPlaceholder')"
              :maxlength="MAX_MESSAGE_LENGTH"
              :class="{'has-content': newMessage.trim() !== '', 'disabled': newMessageDisabled}"
              @keyup.ctrl.enter="sendPrivateMessage()"
            >
            </textarea>
          </div>
          <div
            class="sub-new-message-row d-flex"
          >
            <div
              v-once
              class="guidelines flex-fill"
              v-html="$t('communityGuidelinesIntro')"
            ></div>
            <button
              class="btn btn-primary"
              :class="{'disabled':newMessageDisabled || newMessage === ''}"
              @click="sendPrivateMessage()"
            >
              {{ $t('sendMessage') }}
            </button>
          </div>
        </div>
      </div>
    </div>
  </div>
</template>

<style lang="scss">
@import '@/assets/scss/colors';
@import '@/assets/scss/variables';

$pmHeaderHeight: 56px;

// Content of Private Message should be always full-size (minus the toolbar/resting banner)

#private-message {
  height: calc(100vh - #{$menuToolbarHeight} -
    var(--banner-gift-promo-height, 0px) -
    var(--banner-damage-paused-height, 0px) -
    var(--banner-gems-promo-height, 0px)
  ); // css variable magic :), must be 0px, 0 alone won't work

    .content {
      flex: 1;
      height: calc(100vh - #{$menuToolbarHeight} - #{$pmHeaderHeight} -
        var(--banner-gift-promo-height, 0px) -
        var(--banner-damage-paused-height, 0px) -
        var(--banner-gems-promo-height, 0px)
      );
    }

    .disable-background {
      .toggle-switch-description {
        white-space: nowrap;
        text-overflow: ellipsis;
        overflow: hidden;
        flex: 1;
      }

      .toggle-switch-outer {
        display: flex;
      }

    }

    .modal-body {
      padding: 0rem;
    }

    .modal-content {
      width: 66vw;
    }

    .modal-dialog {
      margin: 10vh 15vw 0rem;
    }

    .modal-header {
      padding: 1rem 0rem;

      .close {
        cursor: pointer;
        margin: 0rem 1.5rem;
        min-width: 0.75rem;
        padding: 0rem;
        width: 0.75rem;
      }
    }

    .toggle-switch-description {
      font-size: 14px;
      font-weight: bold;
      font-style: normal;
      font-stretch: normal;
      line-height: 1.43;
      letter-spacing: normal;
      color: $gray-50;
    }
  }
</style>

<style lang="scss" scoped>
  @import '@/assets/scss/colors.scss';
  @import '@/assets/scss/tiers.scss';
  @import '@/assets/scss/variables.scss';

  $pmHeaderHeight: 56px;
  $background: $white;

  .header-bar {
    height: 56px;
    background-color: $white;
    padding-left: 1.5rem;
    padding-right: 1.5rem;
    align-items: center;

    .mail-icon {
      width: 32px;
      height: 24px;
      object-fit: contain;
    }

    .mail-icon-label {
      margin-bottom: 0;
    }

    .placeholder.svg-icon {
      width: 32px;
    }
  }

  .full-height {
    height: 100%;
    display: flex;
    flex-direction: column;
    justify-content: center;
  }

  .user-link {
    margin-left: 12px;
  }

  .selected-conversion {
    justify-content: center;
    align-items: center;
  }

  #private-message {
    background-color: $background;
    position: relative;
  }

  .disable-background {
    .toggle-switch-description {
      white-space: nowrap;
      text-overflow: ellipsis;
      overflow: hidden;
      flex: 1;
    }

    .toggle-switch-outer {
      display: block !important;
    }

    .toggle-switch {
      float: right !important;
    }

  }

  .modal-body {
    padding: 0rem;
  }

  .modal-content {
    width: 66vw;
  }

  .modal-dialog {
    margin: 10vh 15vw 0rem;
  }

  .modal-header {
    padding: 1rem 0rem;

    .close {
      cursor: pointer;
      margin: 0rem 1.5rem;
      min-width: 0.75rem;
      padding: 0rem;
      width: 0.75rem;
    }
  }

  .toggle-switch-description {
    font-size: 14px;
    font-weight: bold;
    font-style: normal;
    font-stretch: normal;
    line-height: 1.43;
    letter-spacing: normal;
    color: $gray-50;
  }

  .empty-messages {
    flex-flow: column;
    justify-content: center;

    h3, p {
      color: $gray-200;
      margin: 0rem;
    }

    h2 {
      color: $gray-200;
      margin-bottom: 1rem;
    }

    .no-messages-box {
      display: flex;
      flex-direction: column;
      align-items: center;
      width: 330px;
    }

    .envelope {
      color: $gray-400 !important;

      svg {
        width: 86px;
        height: 64px;
      }
    }
  }
</style>

<style lang="scss" scoped>
@import '@/assets/scss/colors';
@import '@/assets/scss/tiers';
@import '@/assets/scss/variables';

$pmHeaderHeight: 56px;
$background: $white;

.header-bar {
  height: 56px;
  background-color: $white;
  align-items: center;

  .left-header {
    padding-left: 1.5rem;
    max-width: 330px;
    align-items: center;
    flex: 1;
  }

  .mail-icon {
    width: 32px;
    height: 24px;
    object-fit: contain;
  }

  .mail-icon-label {
    margin-bottom: 0;
  }

  .placeholder.svg-icon {
    width: 32px;
  }

  .plus-button {
    padding: 10px 14px;

    &.new-message-mode {
      color: $gray-200;
    }
  }
}

.full-height {
  height: 100%;
  display: flex;
  flex-direction: column;
  justify-content: center;
}

.user-link {
  margin-left: 12px;
}

.selected-conversion {
  justify-content: center;
  align-items: center;
}

#private-message {
  background-color: $background;
  position: relative;
}

.disable-background {
  height: 44px;
  background-color: $gray-600;
  padding: 0.75rem 1.5rem;

  border-bottom: 1px solid $gray-500;
}

.disable-background-in-message-list {
  display: flex;
  align-items: center;
  justify-content: center;
  height: 44px;

  color: $yellow-1;
  background: $yellow-500;
  width: 100%;

  .caption {
    font-weight: 700;
    line-height: 24px;
  }

  .text {
    font-weight: 400;
    line-height: 24px;
  }
}

h3 {
  margin: 0rem;

  .svg-icon {
    width: 10px;
    display: inline-block;
    margin-left: .5em;
  }
}

.header-wrap {
  padding: 0.5em;

  h2 {
    margin: 0;
    line-height: 1;
  }
}

.messagePreview {
  display: block;
  display: -webkit-box;
  -webkit-line-clamp: 2;
  -webkit-box-orient: vertical;
  overflow: hidden;
  text-overflow: ellipsis;
  word-break: break-word;
}

.selected-conversion {
  flex: 1;
}

.messages-column {
  flex: 1;
  padding: 0rem;
  display: flex;
  flex-direction: column;

  .empty-messages, .message-scroll {
    flex: 1;
  }
}

.message-scroll {
  overflow-x: hidden;
  padding-top: 0.5rem;

  @media (min-width: 992px) {
    overflow-x: hidden;
    overflow-y: scroll;
  }
}

.full-width {
  width: 100%;
}

.new-message-row {
  width: 100%;
  padding-left: 1.5rem;
  padding-top: 1.5rem;
  padding-right: 1.5rem;

  textarea {
    background: $white;
    display: inline-block;
    vertical-align: bottom;
    border-radius: 2px;
    z-index: 5;

    &.disabled {
      pointer-events: none;
      opacity: 0.64;
      background-color: $gray-500;
    }

    &.has-content {
      --textarea-auto-height: 80px;
    }

    height: var(--textarea-auto-height, 40px);
    min-height: var(--textarea-auto-height, 40px);
    max-height: 300px;
  }
}

.sub-new-message-row {
  padding: 1.5rem;

  .guidelines {
    height: 32px;
    font-size: 12px;
    font-weight: normal;
    font-style: normal;
    font-stretch: normal;
    line-height: 1.33;
    letter-spacing: normal;
    color: $gray-200;
    margin-top: 0.25rem;
    margin-bottom: 0.25rem;
  }

  button {
    height: 32px;
    border-radius: 4px;
    margin-left: 1.5rem;
    white-space: nowrap;

    &.disabled {
      cursor: default;
      pointer-events: none;
      opacity: 0.64;
      background-color: $gray-500;
      color: $gray-100;
    }
  }
}

.sidebar {
  width: 330px;
  background-color: $gray-700;
  padding: 0;
  border-bottom-left-radius: 8px;

  @media only screen and (max-width: 768px) {
    width: 280px;
  }
}

.time {
  font-size: 12px;
  color: $gray-200;
  margin-bottom: 0.5rem;
}

.to-form input {
  width: 60%;
  display: inline-block;
  margin-left: 1em;
}

.empty-sidebar {
  display: flex;
  align-items: center;
}

.floating-message-input {
  background: $background;
  position: fixed;
  bottom: 0;
}

.floating-header-shadow {
  position: absolute;
  top: 0;
  width: 100%;
  height: 56px;
  right: 0;
  z-index: 1;
  pointer-events: none;

  box-shadow: 0 3px 12px 0 rgba(26, 24, 29, 0.24);
}

.center-avatar {
  margin: 0 auto;
}
</style>

<script>
import Vue, { defineComponent } from 'vue';
import moment from 'moment';
import groupBy from 'lodash/groupBy';
import orderBy from 'lodash/orderBy';
import axios from 'axios';
import { MAX_MESSAGE_LENGTH } from '@/../../common/script/constants';
import findIndex from 'lodash/findIndex';
import { mapState } from '@/libs/store';
import styleHelper from '@/mixins/styleHelper';
import toggleSwitch from '@/components/ui/toggleSwitch.vue';
import userLink from '@/components/userLink.vue';

import messageList from '@/components/messages/messageList.vue';
import messageIcon from '@/assets/svg/message.svg?raw';
import mail from '@/assets/svg/mail.svg?raw';
import faceAvatar from '@/components/faceAvatar.vue';
import { EVENTS } from '@/libs/events';
import PmConversationsList from './pm-conversations-list.vue';
import PmEmptyState from './pm-empty-state.vue';
import PmDisabledState from './pm-disabled-state.vue';
import PmNewMessageStarted from './pm-new-message-started.vue';
import StartNewConversationInputHeader from './start-new-conversation-input-header.vue';
import positiveIcon from '@/assets/svg/positive.svg?raw';
import NotificationMixins from '@/mixins/notifications';

// extract to a shared path
const CONVERSATIONS_PER_PAGE = 10;
const PM_PER_PAGE = 50;

const UI_STATES = Object.freeze({
  LOADING: 'LOADING',
  NO_CONVERSATIONS: 'NO_CONVERSATIONS',
  NO_CONVERSATIONS_SELECTED: 'NO_CONVERSATIONS_SELECTED',
  START_NEW_CONVERSATION: 'START_NEW_CONVERSATION',
  CONVERSATION_SELECTED: 'CONVERSATION_SELECTED',
});

export default defineComponent({
  components: {
    StartNewConversationInputHeader,
    PmNewMessageStarted,
    PmDisabledState,
    PmEmptyState,
    PmConversationsList,
    messageList,
    toggleSwitch,
    userLink,
    faceAvatar,
  },
  filters: {
    timeAgo (value) {
      return moment(new Date(value)).fromNow();
    },
  },
  mixins: [styleHelper, NotificationMixins],
  beforeRouteEnter (to, from, next) {
    next(vm => {
      const data = vm.$store.state.privateMessageOptions;

      if ((!data || (data && !data.userIdToMessage)) && vm.$route.query && vm.$route.query.uuid) {
        vm.$store.dispatch('user:userLookup', { uuid: vm.$route.query.uuid }).then(res => {
          if (res && res.data && res.data.data) {
            vm.$store.dispatch('user:newPrivateMessageTo', {
              member: res.data.data,
            });
          }
        });
      } else {
        vm.hasPrivateMessageOptionsOnPageLoad = true;
      }
    });
  },
  data () {
    return {
      icons: Object.freeze({
        messageIcon,
        mail,
        positive: positiveIcon,
      }),
      UI_STATES,
      showStartNewConversationInput: false,
      newConversationTargetUser: null,
      loadingConversations: true,
      showPopover: false,

      /* Conversation-specific data */
      /**
       * @type {PrivateMessages.InitiatedConversation}
       */
      initiatedConversation: null,
      updateConversationsCounter: 0,
      selectedConversation: {},
      conversationPage: 0,
      canLoadMoreConversations: false,
      /** @type {PrivateMessages.ConversationSummaryMessageEntry[]} */
      loadedConversations: [],
      /** @type {Record<string, PrivateMessages.PrivateMessageEntry[]>} */
      messagesByConversation: {}, // cache {uuid: []}

      newMessage: '',
      messages: [],
      messagesLoading: false,
      MAX_MESSAGE_LENGTH: MAX_MESSAGE_LENGTH.toString(),
    };
  },
  computed: {
    ...mapState({ user: 'user.data' }),
    canLoadMore () {
      return this.selectedConversation && this.selectedConversation.canLoadMore;
    },
    conversations () {
      const inboxGroup = groupBy(this.loadedConversations, 'uuid');

      // Add placeholder for new conversations
      if (this.initiatedConversation && this.initiatedConversation.uuid) {
        inboxGroup[this.initiatedConversation.uuid] = [{
          uuid: this.initiatedConversation.uuid,
          user: this.initiatedConversation.user,
          username: this.initiatedConversation.username,
          contributor: this.initiatedConversation.contributor,
          backer: this.initiatedConversation.backer,
          userStyles: this.initiatedConversation.userStyles,
          id: '',
          text: '',
          timestamp: new Date(),
          canReceive: true,
        }];
      }
      // Create conversation objects

      /** @type {PrivateMessages.ConversationEntry[]} */
      const convos = [];

      for (const key in inboxGroup) {
        if (Object.prototype.hasOwnProperty.call(inboxGroup, key)) {
          /**
           * @type {PrivateMessages.ConversationSummaryMessageEntry}
           */
          const recentMessage = inboxGroup[key][0];

          const convoModel = {
            key: recentMessage.uuid,
            name: recentMessage.user,
            // Handles case where from user sent the only message
            // or the to user sent the only message
            username: recentMessage.username,
            date: recentMessage.timestamp,
            lastMessageText: recentMessage.text,
            contributor: recentMessage.contributor,
            userStyles: recentMessage.userStyles,
            backer: recentMessage.backer,
            canReceive: recentMessage.canReceive,
            canLoadMore: false,
            page: 0,
          };

          convos.push(convoModel);
        }
      }

      return convos;
    },
    // Separate from selectedConversation which is not computed
    // so messages don't update automatically
    /* eslint-disable vue/no-side-effects-in-computed-properties */
    selectedConversationMessages () {
      // Vue-subscribe to changes
      const subscribeToUpdate = this.messagesLoading || this.updateConversationsCounter > -1;

      const selectedConversationKey = this.selectedConversation.key;
      const selectedConversation = this.messagesByConversation[selectedConversationKey];
      this.messages = selectedConversation || [];

      const ordered = orderBy(this.messages, [m => m.timestamp], ['asc']);

      if (subscribeToUpdate) {
        return ordered;
      }

      return [];
    },
    filtersConversations () {
      // Vue-subscribe to changes
      const subscribeToUpdate = this.updateConversationsCounter > -1;

      const filtered = subscribeToUpdate && this.conversations;

      const ordered = orderBy(filtered, [o => o.date], ['desc']);

      return ordered;
    },
    placeholderTexts () {
      if (this.user.flags.chatRevoked) {
        return {
          title: this.$t('PMPlaceholderTitleRevoked'),
          description: this.$t('chatPrivilegesRevoked'),
        };
      }
      return {
        title: this.$t('PMPlaceholderTitle'),
        description: this.$t('PMPlaceholderDescription'),
      };
    },

    disabledTexts () {
      if (this.user.flags.chatRevoked) {
        return {
          enableInput: false,
          showBottomInfo: true,
          title: this.$t('PMPlaceholderTitleRevoked'),
          description: this.$t('chatPrivilegesRevoked'),
        };
      }

      if (this.selectedConversation?.key) {
        if (this.user.inbox.blocks.includes(this.selectedConversation.key)) {
          return {
            enableInput: false,
            showBottomInfo: true,
            title: this.$t('PMDisabledCaptionTitle'),
            description: this.$t('PMUnblockUserToSendMessages'),
          };
        }

        if (!this.selectedConversation.canReceive) {
          return {
            enableInput: false,
            showBottomInfo: true,
            title: this.$t('PMCanNotReply'),
            description: this.$t('PMUserDoesNotReceiveMessages'),
          };
        }
      }

      if (this.user.inbox.optOut) {
        return {
          enableInput: true,
          showBottomInfo: false,
          title: this.$t('PMDisabledCaptionTitle'),
          description: this.$t('PMDisabledCaptionText'),
        };
      }

      return null;
    },
    optTextSet () {
      if (!this.user.inbox.optOut) {
        return {
          switchDescription: this.$t('PMDisabled'),
          popoverText: this.$t('PMEnabledOptPopoverText'),
        };
      }
      return {
        switchDescription: this.$t('PMDisabled'),
        popoverText: this.$t('PMDisabledOptPopoverText'),
      };
    },
    selectedConversationFaceAvatarClass () {
      if (this.selectedConversation?.contributor) {
        return `tier${this.selectedConversation.contributor.level}`;
      }
      return '';
    },
    newMessageDisabled () {
      if (this.disabledTexts) {
        return !this.disabledTexts.enableInput;
      }

      return [
        UI_STATES.NO_CONVERSATIONS_SELECTED,
        UI_STATES.NO_CONVERSATIONS,
        UI_STATES.LOADING,
      ].includes(this.uiState);
    },
    uiState () {
      if (this.loadingConversations) {
        return UI_STATES.LOADING;
      }

      if (this.loadedConversations.length === 0) {
        return UI_STATES.NO_CONVERSATIONS;
      }

      // Hiding the "Select a conversation on the left" state,
      // and just picking the first conversation once it loads, right away
      // see reload method
      /* if (!this.selectedConversation.key) {
        return UI_STATES.NO_CONVERSATIONS_SELECTED;
      } */

      if (this.selectedConversationMessages.length === 0) {
        return UI_STATES.START_NEW_CONVERSATION;
      }

      return UI_STATES.CONVERSATION_SELECTED;
    },
    shouldShowInputPanel () {
      const currentUiState = this.uiState;

      switch (currentUiState) {
        case UI_STATES.CONVERSATION_SELECTED:
        case UI_STATES.START_NEW_CONVERSATION: {
          return true;
        }

        default: {
          return false;
        }
      }
    },
  },
  async mounted () {
    this.$store.dispatch('common:setTitle', {
      section: this.$t('messages'),
    });
    // notification click to refresh
    this.$root.$on(EVENTS.PM_REFRESH, async () => {
      await this.reload();
    });

    // header sync button
    this.$root.$on(EVENTS.RESYNC_COMPLETED, async () => {
      await this.reload();
    });

    await this.reload();

    // close modal if the Private Messages page is opened in an existing tab
    this.$root.$emit('bv::hide::modal', 'profile');
    this.$root.$emit('bv::hide::modal', 'members-modal');

    const data = this.$store.state.privateMessageOptions;
    if (data && data.userIdToMessage) {
      this.initiatedConversation = {
        uuid: data.userIdToMessage,
        user: data.displayName,
        username: data.username,
        backer: data.backer,
        contributor: data.contributor,
        userStyles: data.userStyles,
        canReceive: true,
      };

      this.$store.state.privateMessageOptions = {};

      this.selectConversation(this.initiatedConversation.uuid);
    }
  },
  beforeDestroy () {
    this.$root.$off(EVENTS.RESYNC_COMPLETED);
    this.$root.$off(EVENTS.PM_REFRESH);
  },

  methods: {
    async reload () {
      this.loadingConversations = true;
      this.conversationPage = 0;

      this.loadedConversations = [];
      this.selectedConversation = {};

      await this.loadConversations();

      await this.$store.dispatch('user:markPrivMessagesRead');

      await this.selectFirstConversation();

      this.loadingConversations = false;
    },
    async loadConversations () {
      const query = [
        '/api/v4/inbox/conversations',
        `?page=${this.conversationPage}`,
      ];
      this.conversationPage += 1;

      const conversationRes = await axios.get(query.join(''));
      const loadedConversations = conversationRes.data.data;
      this.canLoadMoreConversations = loadedConversations.length === CONVERSATIONS_PER_PAGE;
      this.loadedConversations.push(...loadedConversations);
    },
    messageLiked (message) {
      const messages = this.messagesByConversation[this.selectedConversation.key];

      const chatIndex = findIndex(messages, chatMessage => chatMessage.id === message.id);
      messages.splice(chatIndex, 1, message);
    },
    messageRemoved (message) {
      const messages = this.messagesByConversation[this.selectedConversation.key];

      const messageIndex = messages.findIndex(msg => msg.id === message.id);
      if (messageIndex !== -1) messages.splice(messageIndex, 1);
      if (this.selectedConversationMessages.length === 0) {
        this.initiatedConversation = {
          uuid: this.selectedConversation.key,
          user: this.selectedConversation.name,
          username: this.selectedConversation.username,
          backer: this.selectedConversation.backer,
          contributor: this.selectedConversation.contributor,
        };
      }
    },
    toggleOpt () {
      this.$store.dispatch('user:togglePrivateMessagesOpt');
    },
    async selectConversation (key, forceLoadMessage = false) {
      const convoFound = this.conversations.find(conversation => conversation.key === key);

      this.selectedConversation = convoFound || {};

      if (!this.messagesByConversation[this.selectedConversation.key] || forceLoadMessage) {
        await this.loadMessages();
      }

      this.scrollToBottom();
    },
    sendPrivateMessage () {
      if (!this.newMessage) return;

      const messages = this.messagesByConversation[this.selectedConversation.key];

      messages.push({
        sent: true,
        text: this.newMessage,
        timestamp: new Date(),
        toUser: this.selectedConversation.name,
        toUserName: this.selectedConversation.username,
        toUserContributor: this.selectedConversation.contributor,
        toUserBacker: this.selectedConversation.backer,
        toUUID: this.selectedConversation.uuid,

        id: '-1', // will be updated once the result is back
        likes: {},
        ownerId: this.user._id,
        uuid: this.user._id,
        fromUUID: this.user._id,
        user: this.user.profile.name,
        username: this.user.auth.local.username,
        contributor: this.user.contributor,
        backer: this.user.backer,
        canReceive: true,
      });

      // Remove the placeholder message
      if (this.initiatedConversation
        && this.initiatedConversation.uuid === this.selectedConversation.key) {
        this.loadedConversations.unshift(this.initiatedConversation);
        this.initiatedConversation = null;
      }

      this.selectedConversation.lastMessageText = this.newMessage;
      this.selectedConversation.date = new Date();

      this.$store.dispatch('members:sendPrivateMessage', {
        toUserId: this.selectedConversation.key,
        message: this.newMessage,
      }).then(response => {
        const newMessage = response.data.data.message;
        const messageToReset = messages[messages.length - 1];

        // just set the id, all other infos already set
        messageToReset.id = newMessage.id;
        messageToReset.text = newMessage.text;
        messageToReset.uniqueMessageId = newMessage.uniqueMessageId;

        Object.assign(messages[messages.length - 1], messageToReset);
        this.updateConversationsCounter += 1;
      });

      this.newMessage = '';

      setTimeout(() => {
        this.scrollToBottom();
      }, 150);
    },
    /**
     * This method does a couple of things:
     * - first round:
     *    - tries to scroll down
     *    - in the next tick it triggers it again
     *      (during testing it seemed that the first trigger still had some space left to scroll)
     * - 2nd round:
     *    - tries to scroll down
     *    - in the next tick it checks if the scrollTop is to most it can scroll down,
     *        if it is, it stops from doing that again
     *        if not, it goes into the next round
     * - if we reach round 6 it stops completely,
     *    no need to have a endless loop of just scrolling down
     */
    scrollToBottom (callCount = 0) {
      if (callCount > 5) {
        return;
      }

      if (!this.$refs.chatscroll) {
        // if the message list component not loaded yet, but scrollToBottom was called
        // just try again at a later time
        setTimeout(() => {
          this.scrollToBottom(callCount + 1);
        }, 125);
        return;
      }

      const chatscrollEl = this.$refs.chatscroll.$el;
      // chatscrollBeforeTick.scrollTop = chatscrollBeforeTick.scrollHeight;
      chatscrollEl.scrollTo(0, chatscrollEl.scrollHeight);

      Vue.nextTick(() => {
        if (!this.$refs.chatscroll) {
          return;
        }

        let shouldRetrigger = true;

        if (callCount > 1) {
          const maxPossibleScrollPos = chatscrollEl.scrollHeight - chatscrollEl.clientHeight;

          if (chatscrollEl.scrollTop === maxPossibleScrollPos) {
            shouldRetrigger = false;
          }
        }

        if (shouldRetrigger) {
          setTimeout(() => {
            this.scrollToBottom(callCount + 1);
          }, 125);
        }
      });
    },
    infiniteScrollTrigger () {
      // show loading and wait until the loadMore debounced
      // or else it would trigger on every scrolling-pixel (while not loading)
      if (this.canLoadMore) {
        this.messagesLoading = true;
      }

      return this.loadMore();
    },
    loadMore () {
      this.selectedConversation.page += 1;
      return this.loadMessages();
    },
    async loadMessages () {
      this.messagesLoading = true;

      // use local vars if the loading takes longer
      // and the user switches the conversation while loading
      const conversationKey = this.selectedConversation.key;

      const requestUrl = `/api/v4/inbox/paged-messages?conversation=${conversationKey}&page=${this.selectedConversation.page}`;
      const res = await axios.get(requestUrl);
      const loadedMessages = res.data.data;

      /* eslint-disable max-len */
      this.messagesByConversation[conversationKey] = this.messagesByConversation[conversationKey] || [];
      /* eslint-disable max-len */
      const loadedMessagesToAdd = loadedMessages
        .filter(m => this.messagesByConversation[conversationKey].findIndex(mI => mI.id === m.id) === -1);
      this.messagesByConversation[conversationKey].push(...loadedMessagesToAdd);

      // only show the load more Button if the max count was returned
      this.selectedConversation.canLoadMore = loadedMessages.length === PM_PER_PAGE;
      this.messagesLoading = false;
    },
    async selectFirstConversation () {
      if (this.loadedConversations.length > 0) {
        await this.selectConversation(this.loadedConversations[0].uuid, true);
      }
    },
    triggerStartNewConversationState () {
      this.showStartNewConversationInput = true;
    },
    async startConversationByUsername (targetUserName) {
      // check if the target user exists in current conversations, select that conversation
      /** @type {PrivateMessages.ConversationSummaryMessageEntry} */
      const foundConversation = this.loadedConversations.find(c => c.username === targetUserName);

      if (foundConversation) {
        this.selectConversation(foundConversation.uuid);
        this.showStartNewConversationInput = false;

        return;
      }

      let loadedMember = null;

      try {
        loadedMember = await this.$store.dispatch('members:fetchMemberByUsername', {
          username: targetUserName,
        });
      } catch {
        loadedMember = null;
      }

      if (!loadedMember) {
        this.error(this.$t('targetUserNotExist', { userName: targetUserName }));
        return;
      }

      const loadedMemberUUID = loadedMember.id;

      this.showStartNewConversationInput = false;

      // otherwise create a dummy conversation, load messages for that user
      /**
       * @type {PrivateMessages.ConversationSummaryMessageEntry}
       */
      const newConversationItem = {
        uuid: loadedMemberUUID,
        user: loadedMember.profile.name,
        username: loadedMember.auth.local.username,
        contributor: loadedMember.contributor,
        userStyles: loadedMember,
        canReceive: loadedMember.inbox.canReceive,
        timestamp: new Date(),
        count: 0,
        text: '',
      };

      this.loadedConversations.splice(0, 0, newConversationItem);

      this.selectConversation(loadedMemberUUID);

      if (this.messagesByConversation[loadedMemberUUID]) {
        const messageLengthByConversation = this.messagesByConversation[loadedMemberUUID].length;

        // if messages already exists, update the sidebar entry last message
        if (messageLengthByConversation > 0) {
          /** @type {PrivateMessages.PrivateMessageEntry} */
          const lastMessage = this.messagesByConversation[loadedMemberUUID][messageLengthByConversation - 1];

          newConversationItem.lastMessageText = lastMessage.text;

          return;
        }
      }

      this.newConversationTargetUser = loadedMember;
    },
  },
});
</script>
